基于Gradle的Android应用打包实践

0x01 基本项目结构

使用Android Studio创建的Android项目会划分成三个层级:

  • project : settings.gradle定义了构建应用时包含了哪些模块;build.gradle定义了适用于项目中所有模块的构建配置

  • module : 可以是一个app类型的module,对应生成apk应用;也可以是一个lib类型的module,对应生成aar包. 每个module中包含的build.gradle定义了针对该module的各种构建选项

  • sourceset : 每个module的源码和资源分组为多个源集,main源集包含了所有变体共用的源码和资源. 源集结合productFlavors和buildTypes编译可以很好的实现多套源码,资源,以及依赖库的应用打包

      +--- Comi/
      |
      \--- build.gradle
      \--- setting.gradle
      +--- app/
      |    |
      |    \--- build.gradle
      |    \--- build/
      |    \--- libs/
      |    \--- src/
      |          |
      |          \--- main/
      |          |      \--- java/
      |          |            \--- com.icomico.comi/
      |          |      \--- res/
      |          |            \--- drawable/
      |          |            \--- layout/
      |          |            \--- ...
      |          |      \--- AndroidManifest.xml
      |          \--- jp/
      |          \--- ...
      +--- comi_base/
      +--- ...
    

0x02 构建流程

构建流程涉及到了将项目转化成apk包的各方面,并配合有可灵活配置的构建选项,基本的构建流程包含了以下几步:

  1. 编译器将源码转换成dex文件,并将其它所有内容转换成已编译资源
  2. 打包工具将dex文件和已编译资源合并成单个apk文件,apk文件此时处于未签名状态
  3. 打包工具根据当前构建任务对应buildTypes中选择的signingConfigs选项,使用相应秘钥对apk文件对apk进行签名
  4. 打包工具使用zipalign工具对apk包进行优化
  5. 构建结束,生成最终apk文件
basic build process

0x03 灵活配置构建选项

基于Gradle的Android插件提供了一系列可自定义配置的构建选项,可以方便在构建过程中选择多样的构建方式,以下列出一些常用的构建选项

  • defaultConfig
    包含了用于所有变体种的一些构建选项,如果productFlavors中配置了同样的构建选项,则会覆盖defaultConfig中的对应项

      defaultConfig {
          applicationId 'com.icomico.comi'  //应用包名
          minSdkVersion 10
          targetSdkVersion 23
          versionCode 1000
          versionName '1.0'
          multiDexEnabled true  //支持多dex文件,用于解决方法数超过65K的限制时打包输出问题
      }
    
  • signingConfigs
    包含了app签名文件的各项配置

      signingConfigs {
          release {    
              storeFile file('xxx')
              storePassword 'xxx'
              keyAlias 'xxx'
              keyPassword 'xxx'
              v2SigningEnabled false  //是否使用7.0版本引入的APK signature scheme v2签名方式,默认为true
          }
          debug {
              ...
          }
      }
    
  • buildTypes
    构建类型中可定义多个构建类型,每个构建类型可以选择不同的构建属性,包括使用的签名配置,优化选项,混淆选项等.必须定义至少一个构建类型才能开始构建应用

      buildTypes {
          release {
              signingConfig signingConfigs.release  //使用的签名配置
              minifyEnabled true  //是否使用proguard进行代码压缩优化
              shrinkResources true  //是否进行资源压缩,使用资源压缩前需要使用代码压缩
              proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro', 'proguard-fresco.pro'  //指定使用的proguard配置文件
              manifestPlaceholders = [DEBUG: false]  //替换manifest文件中标记为${DEBUG}的文本
          }
          debug {
              ...
          }
      }
    
  • productFlavors
    谷歌翻译为产品风味的选项,用来发布不同的应用版本,并可以为不同版本指定包括包名,版本号等不同的构建属性

      productFlavors {
          develop {
              applicationId 'com.icomico.comi.dev'  //应用包名
              manifestPlaceholders = [TEMP_CH_NAME name]}  //使用flavor名称替换manifest中的渠道号字段,实现修改渠道号
          official {
              applicationId 'com.icomico.comi'
              manifestPlaceholders = [TEMP_CH_NAME: name]
          }
          ...
      }
    
  • sourceSets
    源集可指定了不同变体使用的源码和资源文件,默认情况下,源集与src目录下的目录结构一一对应,每个src下的目录为一个源集.sourceSets配置项可在变体基础上指定变体使用的源集文件位置,这样可以在定义了大量变体,同时使用的源码和资源文件没有区别的情况下,避免在src下生成同等数量的目录

      sourceSets { 
          develop {
              jni.srcDir 'src/main/jni'
              java.srcDir 'src/dev/java'
              res.srcDir 'src/dev/res'
              manifest.srcFile 'src/dev/AndroidManifest.xml'
          }
          official {
              ...
          }
          ...
      }
    
  • dependencies
    依赖项用以管理来自本地或远程地址上的项目依赖,包括了模块依赖,远程和本地的库文件依赖. 同时可以指定使用依赖项的方式以及为不同变体指定不同的依赖项

      dependencies {
          compile project(':comi_base')  //编译时依赖,gradle 将此配置的依赖项添加到类路径和应用的apk
          apk files('libs/apk_use.jar')  //仅运行时依赖,需要将其与apk一起打包.只可用于jar包形式的依赖项
          provided files('libs/apk_unuse.jar')  //不与apk一起打包的编译时依赖,只可用于jar包形式的依赖项
          officialCompile project(':comi_player')  //构建official版本时使用的依赖项
          debugCompile 'com.squareup.leakcanary:leakcanary-android:1.3'  //构建debug编译类型时使用的依赖项
          releaseCompile 'com.squareup.leakcanary:leakcanary-android-no-op:1.3'  //构建release编译类型时使用的依赖项
          baiduCompile files('libs/BDAutoUpdateSDK_20150605_V1.2.0.jar')  //构建baidu渠道版本时使用的依赖项
          baiduCompile files('libs/need_lib.jar')
          baiduCompile files('libs/patchupdate.jar')
      }
    

0x04 多种变体的打包实践

  • 基于module划分的项目结构
    利用module将项目工程划分为多个app module和多个lib module的结构,多个app工程可以各自选择需要使用到的lib工程,每个lib工程实现相对独立的功能模块.

      +--- Comi/
      |
      +--- app/
      |    |
      |    \--- build.gradle
      |    \--- src/
      |          |
      |          \--- main/
      |          |      \--- java/
      |          |            \--- com.icomico.comi/
      |          |      \--- res/
      |          \--- jp/
      |          |      \--- java/
      |          |            \--- com.icomico.comi/
      |          |                  |
      |          |                  \--- ChConfig.java
      |          \--- official/
      |          |      \--- java/
      |          |            \--- com.icomico.comi/
      |          |                  |
      |          |                  \--- ChConfig.java
      |          \--- ...
      |    \ ...
      +--- app_special/
      +--- comi_base/
      +--- comi_reader/
      +--- comi_player/
      +--- comi_web/
      +--- comi_danmaku/
      +--- ...
    
  • 使用build.gradle脚本管理module的构建配置

      android {
          defaultConfig { ... }
          buildTypes { ... }
          signingConfigs { ... }
          sourceSets { ... }
          productFlavors {
              develop {
                  applicationId 'com.icomico.comi.dev'  //应用包名
                  manifestPlaceholders = [TEMP_CH_NAME name]}  //使用flavor名称替换manifest中的渠道号字段,实现修改渠道号
              official {
                  applicationId 'com.icomico.comi'
                  manifestPlaceholders = [TEMP_CH_NAME: name]
              }
              ...
          }
          ...
      }
      dependencies {
          compile project(':comi_base')
          officialCompile project(':comi_player')
          ...
      }
    
  • 变体
    由buildTypes, productFlavors, sourceSets三项配置, 可以形成gradle打包任务中的多种变体, gradle针对每种变体组合都创建了一个构建任务assemble<productFlavor><buildType>,可指定执行不同变体的构建任务

  • 为变体指定包名,替换指定字段,指定依赖项等
    在app工程的build.gradle构建脚本中,使用productFlavors来定义了多个渠道的变体,并为不同渠道设置了不同的应用包名,渠道号等.同时可以在dependencies中为不同的变体选择了依赖的lib工程,如comi_player库仅为official渠道的变体集成

  • 不同变体使用多套代码及资源
    针对代码,在app工程的src目录下,利用sourceset划分出official和jp两个源集,其中各自定义了用一份java类ChConfig.java,在打包中不同源集的apk文件包含的便是不同的代码文件.main目录下的代码文件会包含在每个源集中,因此在main目录下就不能定义同样的ChConfig.java文件
    针对资源,与代码文件相同,可以在源集目录下定义同名的资源,生成apk包时会覆盖main目录下的同名资源

0x05 渠道打包

  • 结合持续集成环境,可以将渠道号等参数以一定规则作为文件名,在已生成好的apk文件中的META-INF目录下写入空文件,以提升渠道包打包速度

0x06 参考文献

  1. https://developer.android.google.cn/studio/build/index.html
  2. http://tech.meituan.com/mt-apk-packaging.html
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,569评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,499评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,271评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,087评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,474评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,670评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,911评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,636评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,397评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,607评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,093评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,418评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,074评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,092评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,865评论 0 196
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,726评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,627评论 2 270

推荐阅读更多精彩内容